<script>
import {
  GlDropdown,
  GlTooltipDirective,
  GlIcon,
  GlLink,
  GlSprintf,
  GlBadge,
  GlAvatar,
  GlAvatarLink,
} from '@gitlab/ui';
import { isEmpty } from 'lodash';
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
import { __, s__, sprintf } from '~/locale';
import CiIcon from '~/vue_shared/components/ci_icon.vue';
import CommitComponent from '~/vue_shared/components/commit.vue';
import TooltipOnTruncate from '~/vue_shared/components/tooltip_on_truncate/tooltip_on_truncate.vue';
import timeagoMixin from '~/vue_shared/mixins/timeago';
import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
import eventHub from '../event_hub';
import ActionsComponent from './environment_actions.vue';
import DeleteComponent from './environment_delete.vue';
import ExternalUrlComponent from './environment_external_url.vue';
import MonitoringButtonComponent from './environment_monitoring.vue';
import PinComponent from './environment_pin.vue';
import RollbackComponent from './environment_rollback.vue';
import StopComponent from './environment_stop.vue';
import TerminalButtonComponent from './environment_terminal_button.vue';

/**
 * Environment Item Component
 *
 * Renders a table row for each environment.
 */

export default {
  components: {
    ActionsComponent,
    CommitComponent,
    ExternalUrlComponent,
    GlDropdown,
    GlBadge,
    GlIcon,
    GlLink,
    GlSprintf,
    MonitoringButtonComponent,
    PinComponent,
    DeleteComponent,
    RollbackComponent,
    StopComponent,
    TerminalButtonComponent,
    TooltipOnTruncate,
    GlAvatar,
    GlAvatarLink,
    CiIcon,
  },
  directives: {
    GlTooltip: GlTooltipDirective,
  },
  mixins: [timeagoMixin, glFeatureFlagsMixin()],

  props: {
    model: {
      type: Object,
      required: true,
    },

    tableData: {
      type: Object,
      required: true,
    },
  },

  computed: {
    deployIconName() {
      return this.model.isDeployBoardVisible ? 'chevron-down' : 'chevron-right';
    },
    /**
     * Verifies if `last_deployment` key exists in the current Environment.
     * This key is required to render most of the html - this method works has
     * an helper.
     *
     * @returns {Boolean}
     */
    hasLastDeploymentKey() {
      if (this.model && this.model.last_deployment && !isEmpty(this.model.last_deployment)) {
        return true;
      }
      return false;
    },

    /**
     * @returns {Object|Undefined} The `upcoming_deployment` object if it exists.
     * Otherwise, `undefined`.
     */
    upcomingDeployment() {
      return this.model?.upcoming_deployment;
    },

    /**
     * @returns {String} Text that will be shown in the tooltip when
     * the user hovers over the upcoming deployment's status icon.
     */
    upcomingDeploymentTooltipText() {
      return sprintf(s__('Environments|Deployment %{status}'), {
        status: this.upcomingDeployment.deployable.status.text,
      });
    },

    /**
     * Checkes whether the row displayed is a folder.
     *
     * @returns {Boolean}
     */

    isFolder() {
      return this.model.isFolder;
    },

    /**
     * Checkes whether the environment is protected.
     * (`is_protected` currently only set in EE)
     *
     * @returns {Boolean}
     */
    isProtected() {
      return this.model && this.model.is_protected;
    },

    /**
     * Returns whether the environment can be stopped.
     *
     * @returns {Boolean}
     */
    canStopEnvironment() {
      return this.model && this.model.can_stop;
    },

    /**
     * Returns whether the environment can be deleted.
     *
     * @returns {Boolean}
     */
    canDeleteEnvironment() {
      return Boolean(this.model && this.model.can_delete && this.model.delete_path);
    },

    /**
     * Verifies if the `deployable` key is present in `last_deployment` key.
     * Used to verify whether we should or not render the rollback partial.
     *
     * @returns {Boolean|Undefined}
     */
    canRetry() {
      return (
        this.model &&
        this.hasLastDeploymentKey &&
        this.model.last_deployment &&
        this.model.last_deployment.deployable &&
        this.model.last_deployment.deployable.retry_path
      );
    },

    /**
     * Verifies if the autostop date is present.
     *
     * @returns {Boolean}
     */
    canShowAutoStopDate() {
      if (!this.model.auto_stop_at) {
        return false;
      }

      const autoStopDate = new Date(this.model.auto_stop_at);
      const now = new Date();

      return now < autoStopDate;
    },

    /**
     * Human readable deployment date.
     *
     * @returns {String}
     */
    autoStopDate() {
      if (this.canShowAutoStopDate) {
        return {
          formatted: this.timeFormatted(this.model.auto_stop_at),
          tooltip: this.tooltipTitle(this.model.auto_stop_at),
        };
      }
      return {
        formatted: '',
        tooltip: '',
      };
    },

    /**
     * Verifies if the deployment date is present.
     *
     * @returns {Boolean|Undefined}
     */
    canShowDeploymentDate() {
      return this.model && this.model.last_deployment && this.model.last_deployment.deployed_at;
    },

    /**
     * Human readable deployment date.
     *
     * @returns {String}
     */
    deployedDate() {
      if (this.canShowDeploymentDate) {
        return {
          formatted: this.timeFormatted(this.model.last_deployment.deployed_at),
          tooltip: this.tooltipTitle(this.model.last_deployment.deployed_at),
        };
      }
      return {
        formatted: '',
        tooltip: '',
      };
    },

    actions() {
      if (!this.model || !this.model.last_deployment) {
        return [];
      }

      const { manualActions, scheduledActions } = convertObjectPropsToCamelCase(
        this.model.last_deployment,
        { deep: true },
      );
      const combinedActions = (manualActions || []).concat(scheduledActions || []);
      return combinedActions.map((action) => ({
        ...action,
        name: action.name,
      }));
    },

    shouldRenderDeployBoard() {
      return this.model.hasDeployBoard;
    },

    /**
     * Builds the string used in the user image alt attribute.
     *
     * @returns {String}
     */
    userImageAltDescription() {
      if (
        this.model &&
        this.model.last_deployment &&
        this.model.last_deployment.user &&
        this.model.last_deployment.user.username
      ) {
        return sprintf(__("%{username}'s avatar"), {
          username: this.model.last_deployment.user.username,
        });
      }
      return '';
    },

    /**
     * Same as `userImageAltDescription`, but for the
     * upcoming deployment's user
     *
     * @returns {String}
     */
    upcomingDeploymentUserImageAltDescription() {
      return sprintf(__("%{username}'s avatar"), {
        username: this.upcomingDeployment.user.username,
      });
    },

    /**
     * If provided, returns the commit tag.
     *
     * @returns {String|Undefined}
     */
    commitTag() {
      if (this.model && this.model.last_deployment && this.model.last_deployment.tag) {
        return this.model.last_deployment.tag;
      }
      return undefined;
    },

    /**
     * If provided, returns the commit ref.
     *
     * @returns {Object|Undefined}
     */
    commitRef() {
      if (this.model && this.model.last_deployment && this.model.last_deployment.ref) {
        return this.model.last_deployment.ref;
      }
      return undefined;
    },

    /**
     * If provided, returns the commit url.
     *
     * @returns {String|Undefined}
     */
    commitUrl() {
      if (
        this.model &&
        this.model.last_deployment &&
        this.model.last_deployment.commit &&
        this.model.last_deployment.commit.commit_path
      ) {
        return this.model.last_deployment.commit.commit_path;
      }
      return undefined;
    },

    /**
     * If provided, returns the commit short sha.
     *
     * @returns {String|Undefined}
     */
    commitShortSha() {
      if (
        this.model &&
        this.model.last_deployment &&
        this.model.last_deployment.commit &&
        this.model.last_deployment.commit.short_id
      ) {
        return this.model.last_deployment.commit.short_id;
      }
      return undefined;
    },

    /**
     * If provided, returns the commit title.
     *
     * @returns {String|Undefined}
     */
    commitTitle() {
      if (
        this.model &&
        this.model.last_deployment &&
        this.model.last_deployment.commit &&
        this.model.last_deployment.commit.title
      ) {
        return this.model.last_deployment.commit.title;
      }
      return undefined;
    },

    /**
     * If provided, returns the commit tag.
     *
     * @returns {Object|Undefined}
     */
    commitAuthor() {
      if (
        this.model &&
        this.model.last_deployment &&
        this.model.last_deployment.commit &&
        this.model.last_deployment.commit.author
      ) {
        return this.model.last_deployment.commit.author;
      }

      return undefined;
    },

    /**
     * Verifies if the `retry_path` key is present and returns its value.
     *
     * @returns {String|Undefined}
     */
    retryUrl() {
      if (
        this.model &&
        this.model.last_deployment &&
        this.model.last_deployment.deployable &&
        this.model.last_deployment.deployable.retry_path
      ) {
        return this.model.last_deployment.deployable.retry_path;
      }
      return undefined;
    },

    /**
     * Verifies if the `last?` key is present and returns its value.
     *
     * @returns {Boolean|Undefined}
     */
    isLastDeployment() {
      // name: 'last?' is a false positive: https://gitlab.com/gitlab-org/frontend/eslint-plugin-i18n/issues/26#possible-false-positives
      // Vue i18n ESLint rules issue: https://gitlab.com/gitlab-org/gitlab-foss/issues/63560
      // eslint-disable-next-line @gitlab/require-i18n-strings
      return this.model && this.model.last_deployment && this.model.last_deployment['last?'];
    },

    /**
     * Builds the name of the builds needed to display both the name and the id.
     *
     * @returns {String}
     */
    buildName() {
      if (this.model && this.model.last_deployment && this.model.last_deployment.deployable) {
        const { deployable } = this.model.last_deployment;
        return `${deployable.name} #${deployable.id}`;
      }
      return '';
    },

    /**
     * Builds the needed string to show the internal id.
     *
     * @returns {String}
     */
    deploymentInternalId() {
      if (this.model && this.model.last_deployment && this.model.last_deployment.iid) {
        return `#${this.model.last_deployment.iid}`;
      }
      return '';
    },

    /**
     * Same as `deploymentInternalId`, but for the upcoming deployment
     *
     * @returns {String}
     */
    upcomingDeploymentInternalId() {
      return `#${this.upcomingDeployment.iid}`;
    },

    /**
     * Verifies if the user object is present under last_deployment object.
     *
     * @returns {Boolean}
     */
    deploymentHasUser() {
      return (
        this.model &&
        !isEmpty(this.model.last_deployment) &&
        !isEmpty(this.model.last_deployment.user)
      );
    },

    /**
     * Returns the user object nested with the last_deployment object.
     * Used to render the template.
     *
     * @returns {Object}
     */
    deploymentUser() {
      if (
        this.model &&
        !isEmpty(this.model.last_deployment) &&
        !isEmpty(this.model.last_deployment.user)
      ) {
        return this.model.last_deployment.user;
      }
      return {};
    },

    /**
     * Checkes whether to display no deployment text.
     *
     * @returns {Boolean}
     */
    showNoDeployments() {
      return !this.hasLastDeploymentKey && !this.isFolder;
    },

    /**
     * Verifies if the build name column should be rendered by verifing
     * if all the information needed is present
     * and if the environment is not a folder.
     *
     * @returns {Boolean}
     */
    shouldRenderBuildName() {
      return (
        !this.isFolder &&
        !isEmpty(this.model.last_deployment) &&
        !isEmpty(this.model.last_deployment.deployable)
      );
    },

    /**
     * Verifies the presence of all the keys needed to render the buil_path.
     *
     * @return {String}
     */
    buildPath() {
      if (
        this.model &&
        this.model.last_deployment &&
        this.model.last_deployment.deployable &&
        this.model.last_deployment.deployable.build_path
      ) {
        return this.model.last_deployment.deployable.build_path;
      }

      return '';
    },

    /**
     * Verifies the presence of all the keys needed to render the external_url.
     *
     * @return {String}
     */
    externalURL() {
      return this.model.external_url || '';
    },

    /**
     * Verifies if deplyment internal ID should be rendered by verifing
     * if all the information needed is present
     * and if the environment is not a folder.
     *
     * @returns {Boolean}
     */
    shouldRenderDeploymentID() {
      return (
        !this.isFolder &&
        !isEmpty(this.model.last_deployment) &&
        this.model.last_deployment.iid !== undefined
      );
    },

    environmentPath() {
      return this.model.environment_path || '';
    },

    monitoringUrl() {
      return this.model.metrics_path || '';
    },

    canShowMetricsLink() {
      return Boolean(!this.glFeatures.removeMonitorMetrics && this.monitoringUrl);
    },

    terminalPath() {
      return this.model?.terminal_path ?? '';
    },

    autoStopUrl() {
      return this.model.cancel_auto_stop_path || '';
    },

    displayEnvironmentActions() {
      return (
        this.actions.length > 0 ||
        this.externalURL ||
        this.canShowMetricsLink ||
        this.canStopEnvironment ||
        this.canDeleteEnvironment ||
        this.canRetry
      );
    },

    folderIconName() {
      return this.model.isOpen ? 'chevron-down' : 'chevron-right';
    },

    upcomingDeploymentCellClasses() {
      return [
        this.tableData.upcoming.spacing,
        { 'gl-display-none gl-md-display-block': !this.upcomingDeployment },
      ];
    },
    tableNameSpacingClass() {
      return this.isFolder ? 'section-100' : this.tableData.name.spacing;
    },
    hasExtraActions() {
      return Boolean(
        this.canRetry ||
          this.canShowAutoStopDate ||
          this.canShowMetricsLink ||
          this.terminalPath ||
          this.canDeleteEnvironment,
      );
    },
  },

  methods: {
    toggleDeployBoard() {
      eventHub.$emit('toggleDeployBoard', this.model);
    },
    onClickFolder() {
      eventHub.$emit('toggleFolder', this.model);
    },

    /**
     * Returns the field title that will be shown in the field's row
     * in the mobile view.
     *
     * @returns `field.mobileTitle` if present;
     * if not, falls back to `field.title`.
     */
    getMobileViewTitleForField(fieldName) {
      const field = this.tableData[fieldName];

      return field.mobileTitle || field.title;
    },
  },
};
</script>
<template>
  <div
    :class="{
      'js-child-row environment-child-row': model.isChildren,
      'folder-row': isFolder,
    }"
    class="gl-responsive-table-row"
    role="row"
  >
    <div
      class="table-section section-wrap text-truncate"
      :class="tableNameSpacingClass"
      role="gridcell"
      data-testid="environment-name-cell"
    >
      <div v-if="!isFolder" class="table-mobile-header" role="rowheader">
        {{ getMobileViewTitleForField('name') }}
      </div>

      <span v-if="shouldRenderDeployBoard" class="deploy-board-icon" @click="toggleDeployBoard">
        <gl-icon :name="deployIconName" />
      </span>

      <span
        v-if="!isFolder"
        v-gl-tooltip
        :title="model.name"
        class="environment-name table-mobile-content"
      >
        <a data-qa-selector="environment_link" :href="environmentPath">
          <span v-if="model.size === 1">{{ model.name }}</span>
          <span v-else>{{ model.name_without_type }}</span>
        </a>
        <gl-badge v-if="isProtected" variant="success">{{
          s__('Environments|protected')
        }}</gl-badge>
      </span>
      <span
        v-else
        v-gl-tooltip
        :title="model.folderName"
        class="folder-name"
        role="button"
        @click="onClickFolder"
      >
        <gl-icon :name="folderIconName" class="folder-icon" />

        <gl-icon name="folder" class="folder-icon" />

        <span> {{ model.folderName }} </span>

        <gl-badge>{{ model.size }}</gl-badge>
      </span>
    </div>

    <div
      v-if="!isFolder"
      class="table-section deployment-column d-none d-md-block"
      :class="tableData.deploy.spacing"
      role="gridcell"
      data-testid="environment-deployment-id-cell"
    >
      <span v-if="shouldRenderDeploymentID" class="text-break-word">
        {{ deploymentInternalId }}
      </span>

      <span
        v-if="!isFolder && deploymentHasUser"
        class="text-break-word gl-display-inline-flex gl-align-items-center"
      >
        <gl-sprintf :message="s__('Environments|by %{avatar}')">
          <template #avatar>
            <gl-avatar-link :href="deploymentUser.web_url" class="gl-ml-2">
              <gl-avatar
                :src="deploymentUser.avatar_url"
                :entity-name="deploymentUser.username"
                :title="deploymentUser.username"
                :alt="userImageAltDescription"
                :size="24"
              />
            </gl-avatar-link>
          </template>
        </gl-sprintf>
      </span>

      <div v-if="showNoDeployments" class="commit-title table-mobile-content">
        {{ s__('Environments|No deployments yet') }}
      </div>
    </div>

    <div
      v-if="!isFolder"
      class="table-section d-none d-md-block"
      :class="tableData.build.spacing"
      role="gridcell"
      data-testid="environment-build-cell"
    >
      <a v-if="shouldRenderBuildName" :href="buildPath" class="build-link cgray">
        <tooltip-on-truncate
          :title="buildName"
          truncate-target="child"
          class="flex-truncate-parent"
        >
          <span class="flex-truncate-child">
            {{ buildName }}
          </span>
        </tooltip-on-truncate>
      </a>
    </div>

    <div v-if="!isFolder" class="table-section" :class="tableData.commit.spacing" role="gridcell">
      <div role="rowheader" class="table-mobile-header">
        {{ getMobileViewTitleForField('commit') }}
      </div>
      <div v-if="hasLastDeploymentKey" class="js-commit-component table-mobile-content">
        <commit-component
          :tag="commitTag"
          :commit-ref="commitRef"
          :commit-url="commitUrl"
          :short-sha="commitShortSha"
          :title="commitTitle"
          :author="commitAuthor"
        />
      </div>
    </div>

    <div v-if="!isFolder" class="table-section" :class="tableData.date.spacing" role="gridcell">
      <div role="rowheader" class="table-mobile-header">
        {{ getMobileViewTitleForField('date') }}
      </div>
      <span
        v-if="canShowDeploymentDate"
        v-gl-tooltip
        :title="deployedDate.tooltip"
        class="environment-created-date-timeago table-mobile-content flex-truncate-parent"
      >
        <span class="flex-truncate-child">
          {{ deployedDate.formatted }}
        </span>
      </span>
    </div>

    <div
      v-if="!isFolder"
      class="table-section"
      :class="upcomingDeploymentCellClasses"
      role="gridcell"
      data-testid="upcoming-deployment"
    >
      <div role="rowheader" class="table-mobile-header">
        {{ getMobileViewTitleForField('upcoming') }}
      </div>
      <div
        v-if="upcomingDeployment"
        class="gl-w-full gl-display-flex gl-flex-direction-row gl-md-flex-direction-column! gl-justify-content-end"
        data-testid="upcoming-deployment-content"
      >
        <div class="gl-display-flex gl-align-items-center">
          <span class="gl-mr-2">{{ upcomingDeploymentInternalId }}</span>
          <gl-link
            v-if="upcomingDeployment.deployable"
            v-gl-tooltip
            :href="upcomingDeployment.deployable.build_path"
            :title="upcomingDeploymentTooltipText"
            data-testid="upcoming-deployment-status-link"
          >
            <ci-icon class="gl-mr-2" :status="upcomingDeployment.deployable.status" />
          </gl-link>
        </div>
        <span
          v-if="upcomingDeployment.user"
          class="text-break-word gl-display-inline-flex gl-align-items-center gl-mt-2"
        >
          <gl-sprintf :message="s__('Environments|by %{avatar}')">
            <template #avatar>
              <gl-avatar-link :href="upcomingDeployment.user.web_url" class="gl-ml-2">
                <gl-avatar
                  :src="upcomingDeployment.user.avatar_url"
                  :alt="upcomingDeploymentUserImageAltDescription"
                  :entity-name="upcomingDeployment.user.username"
                  :title="upcomingDeployment.user.username"
                  :size="24"
                />
              </gl-avatar-link>
            </template>
          </gl-sprintf>
        </span>
      </div>
    </div>

    <div v-if="!isFolder" class="table-section" :class="tableData.autoStop.spacing" role="gridcell">
      <div role="rowheader" class="table-mobile-header">
        {{ getMobileViewTitleForField('autoStop') }}
      </div>
      <span
        v-if="canShowAutoStopDate"
        v-gl-tooltip
        :title="autoStopDate.tooltip"
        class="table-mobile-content flex-truncate-parent"
      >
        <span class="flex-truncate-child js-auto-stop">{{ autoStopDate.formatted }}</span>
      </span>
    </div>

    <div
      v-if="!isFolder && displayEnvironmentActions"
      class="table-section table-button-footer"
      :class="tableData.actions.spacing"
      role="gridcell"
    >
      <div class="btn-group table-action-buttons" role="group">
        <external-url-component
          v-if="externalURL"
          :external-url="externalURL"
          data-track-action="click_button"
          data-track-label="environment_url"
        />

        <actions-component
          v-if="actions.length > 0"
          :actions="actions"
          data-track-action="click_dropdown"
          data-track-label="environment_actions"
        />

        <stop-component
          v-if="canStopEnvironment"
          :environment="model"
          class="gl-z-index-2"
          data-track-action="click_button"
          data-track-label="environment_stop"
        />

        <gl-dropdown
          v-if="hasExtraActions"
          icon="ellipsis_v"
          text-sr-only
          :text="__('More actions')"
          category="secondary"
          no-caret
        >
          <rollback-component
            v-if="canRetry"
            :environment="model"
            :is-last-deployment="isLastDeployment"
            :retry-url="retryUrl"
            data-track-action="click_button"
            data-track-label="environment_rollback"
          />

          <pin-component
            v-if="canShowAutoStopDate"
            :auto-stop-url="autoStopUrl"
            data-track-action="click_button"
            data-track-label="environment_pin"
          />

          <monitoring-button-component
            v-if="canShowMetricsLink"
            :monitoring-url="monitoringUrl"
            data-track-action="click_button"
            data-track-label="environment_monitoring"
            data-testid="environment-monitoring"
          />

          <terminal-button-component
            v-if="terminalPath"
            :terminal-path="terminalPath"
            data-track-action="click_button"
            data-track-label="environment_terminal"
          />

          <delete-component
            v-if="canDeleteEnvironment"
            :environment="model"
            data-track-action="click_button"
            data-track-label="environment_delete"
          />
        </gl-dropdown>
      </div>
    </div>
  </div>
</template>
